# **AVR Interfacing**

**Serial Peripheral Interface (SPI)** 

# **Agenda**

- Introduction
- Registers Description
- Programming

- The Serial Peripheral Interface (SPI) is a bus interface connection protocol originally started by Motorola Corp. It uses four pins for communication.
  - MISO (Master In Slave Out): Master receives data and slave transmits data through this pin.
  - MOSI (Master Out Slave In): Master transmits data and slave receives data through this pin.
  - > SCK (Shift Clock): Master generates this clock for the communication, which is used by slave. Only master can initiate serial clock.
  - > SS (Slave Select): Master can select slave through this pin.



- SPI devices have 8-bit shift registers to send and receive data.
  - Whenever master need to send data, it places data on shift register and generate required clock.
  - Whenever master want to read data, slave places data on shift register and master generate required clock.

**Note that** SPI is full duplex communication protocol i.e. data on master and slave shift registers get interchanged at a same time.

The interconnection between master and slave using SPI is shown in below figure.



SPI Master Slave Interconnection

#### Pins Configurations:

| 5 | SPI Pins | Pin No. on<br>ATmega16 chip | Pin Direction in<br>master mode | Pin Direction in slave mode |
|---|----------|-----------------------------|---------------------------------|-----------------------------|
|   | MISO     | 7                           | Input                           | Output                      |
| - | MOSI     | 6                           | Output                          | Input                       |
|   | SCK      | 8                           | Output                          | Input                       |
|   | SS       | 5                           | Output                          | Input                       |

#### AVR ATmega16/32 SPI Pins (ADC0) (XCK/T0) PB0 PAG (T1) PB1 (ADC1) (INT2/AIN0) PB2 (ADC2) (OC0/AIN1) PA3 (ADC3) PB3 (SS) PB4 (ADC4) (ADC5) (MOSI) PB5 (MISO) PB6 **ATmega** PA6 (ADC6) (SCK) PB7 PA7 (ADC7) 16/32 RESET 32 AREF 31 VCC AGND GND 30 AVCC XTAL2 29 PC7 (TOCS2) XTAL1 13 28 PC6 (TOCS1) PC5 27 (TD1) (RXD) PD0 26 (TXD) PD1 PC4 (TD0) 16 25 (INTO) PD2 PC3 (TMS) 24 PC2 (TCK) (INT1) PD3 23 (OC1B) PD4 PC1 (SDA) (OC1A) PD5 19 PC0 (SCL) 20 PD7 (OC2) (ICP1) PD6

- AVR ATmega16/32 uses three registers to configure SPI communication that are SPI Control Register, SPI Status Register and SPI Data Register.
  - SPCR: SPI Control Register



Bit 7 - SPIE: SPI Interrupt Enable bit

**1** = Enable SPI interrupt.

**0** = Disable SPI interrupt.

Bit 6 - SPE: SPI Enable bit

1 = Enable SPL

0 = Disable SPI.

Bit 5 - DORD: Data Order bit

1 = LSB transmitted first.

0 = MSB transmitted first.

Bit 4 - MSTR: Master/Slave Select bit

1 = Master mode.

**0** = Slave Mode.

Bit 3 – CPOL: Clock Polarity Select bit

**1** = Clock start from logical one.

**0** = Clock start from logical zero.

Bit 2 - CPHA: Clock Phase Select bit

**1** = Data sample on trailing clock edge.

**0** = Data sample on leading clock edge.

Bit 1:0 - SPR1: SPR0 SPI Clock Rate Select bits

#### SCK clock frequency select bit setting

| SPI2X | SPR1 | SPR0 | SCK<br>Frequency |
|-------|------|------|------------------|
| 0     | 0    | 0    | Fosc/4           |
| 0     | 0    | 1    | Fosc/16          |
| 0     | 1    | 0    | Fosc/64          |
| 0     | 1    | 1    | Fosc/128         |
| 1     | 0    | 0    | Fosc/2           |
| 1     | 0    | 1    | Fosc/8           |
| 1     | 1    | 0    | Fosc/32          |
| 1     | 1    | 1    | Fosc/64          |

...

| CPOL | Leading Edge | Trailing Edge |
|------|--------------|---------------|
| 0    | Rising       | Falling       |
| 1    | Falling      | Rising        |

20

| СРНА | Leading Edge | Trailing Edge |
|------|--------------|---------------|
| 0    | Sample       | Setup         |
| 1    | Setup        | Sample        |



Figure 2. SPI Mode 0, CPOL = 0, CPHA = 0: CLK idle state = low, data sampled on rising edge and shifted on falling edge.



Figure 3. SPI Mode 1, CPOL = 0, CPHA = 1: CLK idle state = low, data sampled on the falling edge and shifted on the rising edge.



Figure 4. SPI Mode 3, CPOL = 1, CPHA = 1: CLK idle state = high, data sampled on the falling edge and shifted on the rising edge.



Figure 5. SPI Mode 2, CPOL = 1, CPHA = 0: CLK idle state = high, data sampled on the rising edge and shifted on the falling edge.

- AVR ATmega16/32 uses three registers to configure SPI communication that are SPI Control Register, SPI Status Register and SPI Data Register.
  - > SPSR: SPI Status Register



Bit 7 - SPIF: SPI interrupt flag bit

This flag gets set when serial transfer is complete.

Bit 6 - WCOL: Write Collision Flag bit

This bit gets set when SPI data register (SPDR) is written during precious data

transfer.

Bit 5:1 - Reserved Bits

Bit 0 - SPI2X: Double SPI Speed bit

When set, SPI speed (SCK frequency) get doubled.

- AVR ATmega16/32 uses three registers to configure SPI communication that are SPI Control Register, SPI Status Register and SPI Data Register.
  - SPDR: SPI Data Register



- ✓ SPI Data register used to transfer data between Register file and SPI Shift Register.
- ✓ Writing to the SPDR initiates data transmission.
- When device is in master mode
  - Master writes data byte in SPDR. Writing to SPDR starts the data transmission.
  - > 8-bit data starts shifting out towards slave and after the complete byte is shifted, SPI clock generator stops and SPIF bit gets set.
- When device is in slave mode
  - > Slave SPI interface remains in sleep as long as SS pin is held high by master.
  - ➤ It activates only when SS pin is driven low. Data is shifted out with incoming SCK clock from master during write operation.
  - > SPIF is set after complete shifting of a byte.

- SS pin functionality Master mode
  - > In master mode SS pin is used as GPIO pin.
  - ➤ Make SS pin direction as output to select slave device using this pin.
- SS pin functionality Slave mode
  - > In slave mode SS pin is always configured as input.
  - When it is low SPI activates.
  - > And when it driven high SPI logic gets reset and does not receive any incoming data.

- Example:
- Let's do SPI communication using AVR family based ATmega16 (Master) and ATmega32 (Slave). Master will send continuous count to Slave. Display the sent and received data on 16x2 LCD.



#### Example:

Let's do SPI communication using AVR family based ATmega16 (Master) and ATmega32 (Slave). Master will send continuous count to Slave. Display the sent and received data on 16x2 LCD.

Let's first program Master device

> SPI Master Initialization steps

To initialize as Master, do the following steps

- ✓ Make MOSI, SCK and SS pins directions as output.
- ✓ Make MISO pin direction as input.
- ✓ Make SS pin High.
- ✓ Enable SPI in Master mode by setting SPE and MSTR bits in SPCR register.
- ✓ Set SPI Clock Rate Bits combination to define SCK frequency.

- > SPI Master Write steps
  - ✓ Copy data to be transmitted in SPDR register.
  - ✓ Wait until transmission is complete i.e. poll SPIF flag to become High.
  - ✓ While SPIF flag gets set read SPDR using flush buffer.
  - ✓ SPIF bit is cleared by H/W when executing corresponding ISR routine.
  - ✓ Note that to clear SPIF bit, need to read SPIF and SPDR registers alternately.
- ❖SPI\_Write function

Input argument: It has input argument of data to be transmit.

- > SPI Master Read steps
  - ✓ Since writing to SPDR generates SCK for transmission, write dummy data in SPDR register.
  - ✓ Wait until transmission is completed i.e. poll SPIF flag till it becomes High.
  - ✓ While SPIF flag gets set, read requested received data in SPDR.
- ❖ SPI\_Read function

Return value: It returns received char data type.

#### Program for SPI Master device

```
#define F CPU 8000000UL /* Define CPU Frequency 8MHz */
#include <avr/io.h> /* Include AVR std. library file */
#include <util/delay.h> /* Include Delay header file */
#include <stdio.h> /* Include Std. i/p o/p file */
#include <string.h> /* Include String header file */
#include "LCD_16x2_H_file.h" /* Include LCD header file */
#include "SPI Master H file.h" /* Include SPI master header file */
int main (void)
   uint8 t count;
   char buffer[5];
   LCD Init();
   SPI Init();
   LCD String xy(1, 0, "Master Device");
   LCD String xy(2, 0, "Sending Data: ");
   SS Enable;
   count = 0;
                       * Send Continuous count */
   while (1)
       SPI Write (count);
       sprintf(buffer, "%d ", count);
       LCD String xy(2, 13, buffer);
       count++;
       delay ms(500);
```

#### Now Program for Slave device:

- > SPI Slave Initialization steps
  - ✓ Make MOSI, SCK and SS pins direction of device as input.
  - ✓ Make MISO pin direction of device as output.
  - ✓ Enable SPI in slave mode by setting SPE bit and clearing MSTR bit.
- ❖ SPI\_Init function

- > SPI Slave transmit steps
  - ✓ It has same function and steps as we do SPI Write in Master mode.
- > SPI Slave Receive steps
  - ✓ Wait until SPIF becomes High.
  - ✓ Read received data from SPDR register.
- SPI\_Receive function Return value: it returns received char data type.

#### Program for SPI Slave device

```
#include <util/delay.h> /* Include Delay header file */
#include <stdio.h> /* Include std. i/p o/p file */
#include <string.h> /* Include string header file */
#include "LCD 16x2 H file.h" /* Include LCD header file */
#include "SPI Slave H file.h" /* Include SPI slave header file */
int main (void)
   uint8 t count;
   char buffer[5];
   LCD Init();
   SPI Init();
   LCD String xy(1, 0, "Slave Device");
   LCD String xy(2, 0, "Receive Data: ");
   while (1) /* Receive count continuous */
      count = SPI Receive();
      sprintf(buffer, "%d ", count);
      LCD String xy(2, 13, buffer);
```